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Abstract 

In this article we discuss the presentation of a random binary matrix using sequence of whole nonneg- 
ative numbers. We examine some advantages and disadvantages of this presentation as an alternative of 
the standard presentation using two-dimensional array. It is shown that the presentation of binary matri- 
ces using ordered n-tuples of natural numbers makes the algorithms faster and saves a lot of memory. In 
this work we use object-oriented programming using the syntax and the semantic of CH — h programming 
language. 

1 Introduction. 
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One of the basic principles of object-oriented programming is encapsulation, which means that the client 
, knows objects' data, properties and methods, to which events the object reacts, but it is not necessary 

to have the information about their realization and the algorithms used in the member functions of the 
corresponding class. Here we ask the following question: if there are two different classes, which describe 
one and the same object in mathematics or in the real world, and they have one and the same properties 
and methods, the question is which one of these two classes we have to choose. The answer is trivial: the 
class, which objects use less memory and which methods work faster, i.e. which methods use less standard 
computer operations. 

The present work is a continuation of the work [5]. Our aim is to show that the presentation of the 
binary matrices using ordered n-tuple of whole numbers and the bitwise operations make the realization of 
a better class(as it was mentioned above) compared with the standard presentation of the binary matrices 
using two-dimensional n x n array of whole numbers. About the definition and some examples how to use 
bitwise operations see [21 13 El [7] . We recommend [4] about the object-oriented programming using CH — h 
| language in the area of the computer algebra. 

We examine the set B = {0, 1}. B together with the operations conjunction &&, disjunction || and 
negation ! form the all known boolean algebra 23(&&, [|, !), which role is very important in the computers and 
programming. We will use the pointed symbols for the operations in £>(&&, ||, !), have in mind the semantic 
and the syntax of these operations in the widespread CH — V programming language. 
; i ■ Binary matrix (or boolean, or (0,l)-matrix) is a matrix, which elements belong to the set B = {0, 1}. We 

| denote B n the set of all n x n square matrices. 

Let A = (aij) and B = (bij) are matrices of B n . We consider the semantic and syntax of C++ language 
and the first index is 0, i.e i, j £ {0, 1, 2, . . . , n — 1}. We examine the following operations in B n , defined 
according to our aim as follows: 
component conjunction 

(1) A && B = C = (dj) 

where by definition for every i, j € {0, 1, 2 . . . , n — 1} ; = && bij 

component disjunction 

(2) A\\B = C = (cij) 

where by definition for every i, j € {0, 1, 2 . . . , n — 1} dj = aij \\ bij 
component negation 

(3) \A = C= {dj) 

where by definition for every i, j G {0, 1, 2 . . . , n — 1} =!aij 
transpose 

(4) t{A) = C={ Cij ) 

*2000 Mathematics Subject Classification: 68N15, 68W40, 15B34 

tKey words: binary matrix, object-oriented programming, CH — h programming language, bitwise operations, computer 
algebra 



1 



where by definition for every i,j £ {0, 1, 2 
logical product 

(5) 

where by definition for every i,j £ {0, 1, 2 

c n = V && = ( ai0 && I' && ^ J') II " ' 'I (^"-i && bn - 1 i) 

That is the way B n and the above-described operations &&. ||, !, i() and * form the algebra £>„(&&, || , !, t(), *). 
Here and in the whole article the term algebra means abstract algebra considering the definition given in [T], 
and namely set equipped with various operations, assumed to satisfy some specified system of axiomatic 
laws. It is naturally we to put the linear order, and exactly the lexicographic order in £>„(&&, ||, !,£(), *). 

We examine the following set of standard operations with integer arguments in the CH — h programming 
language: 

(6) Op = {+, -, *, /, %, «, », &, |, A, ~, &&, ||, !, =, if, <, <=, >, >=, ==, ! =} 

these operations mean addition, subtraction, multiplication, division, integer division, bitwise left shift, bit- 
wise right shift, bitwise "and", bitwise "or", bitwise "exclusive or", bitwise "negation", conjunction, disjunction, 
negation, assignment, if check, and comparing. We consider that the time needed for each of these operations 
of the set Op are proportional, i.e. if t\ and ti are the times needed to execute two random operations of 
Op, then ii = Ctz, where C is a const. The algorithms in this work are evaluated according to the number 
of the needed operations of the set Op. 

The present work is also an appropriate example how to use bitwise operations in the object-oriented 
programming courses. This matter does not take enough place in the studying literature(see for example 
0). 

2 Two classes, which describe the algebra B n (8z8z, ||, !,£(), *). 

To create the first class we use the standard realization of the binary matrices: using two-dimensional n x n 
array of whole numbers and standard algorithms to execute the operations (jlj ~ © ■ Let denote this class 
Bn_array. 

A square binary n x n matrix, as it is described in [5] can be realized using ordered n-tuple of whole 
nonnegative numbers, which belong to the closed interval [0, 2™ — 1]. There is one to one correspondence 
between the representation of the integers in decimal and in binary number system. Let denote Bn_tuple 
the class which describes the algebra ||, !, t(), *) using ordered n-tuple of whole nonnegative numbers. 

These two classes have one and the same specifications (we use the terminology in [H[5]), we will describe 
these specification using the name Bn_X, i.e. "X" means "array", or "tuple" depending on the case. 

Let the two classes Bn_X have the following specification: 

class Bn_X { 
int n; 
int *Matr; 
public : 

/* constructor without parameter: */ 
Bn_X() ; 

/* constructor with parameter n pointing the row of the square matrix: */ 

Bn_X (unsigned int) ; 
/* copy constructor: */ 

Bn_X (const Bn_X &) ; 
/* destructor: */ 

~Bn_X() ; 

/* predefines the assignment operator: */ 

Bn_X & operator = (const Bn_X &) ; 
/* returns the size (row) of the matrix: */ 

int get_n() { return n; }; 
/* sets value 1 to the element (i,j) of the matrix: */ 

void set_l (int, int); 
/* sets value to the element (i,j) of the matrix: */ 



...,n—lj Cij = dji 

A*B = C= (dj) 
...,n-l} 
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void set_0 (int,int); 
/* fills row i of the matrix using integer number r 
(just for the class Bn_tuple) : */ 

void set_row(int , int) ; // must not be included when "X" = "array"! 
/* gets element (i,j) of the matrix: */ 

int get_element (int, int); 
/* gets the row i of the matrix 
(just for the class Bn_tuple) : */ 

int get_row (int); // must not be included when "X" = "array"! 
/* transposes matrix : */ 

Bn_X t () ; 

/* predefines operators according to (1), (2), (3) z (5): */ 

Bn_X operator && (Bn_X &) ; 

Bn_X operator I I (Bn_X &) ; 

Bn_X operator ! () ; 

Bn_X operator * (Bn_X &) ; 
/* defines order (lexicographical) */ 

int operator < (Bn_X &) ; 

> 

When we predefine operators &&, || and *, if the dimensions of the two operands are not equal, then we 
receive the zero matrix of order the same as the first operand. But this result is not correct. We can say 
something more: in this case the operation is not defined, i.e. the result of the operation function is not 
correct and we have to be very careful in such situations. The situation is the same about the entered linear 
order, i.e. although the lexicographic order can be put for the words with different length, we examine by 
definition only the matrices of one and the same order. The result we receive when we compare the matrices 
with different dimensions is not correct. We give suitable massages in this situations. 

Since the objects of the two classes get dynamic operation memory, using pointers and the new operator, 
then to work the applications, using such objects, it is necessary to predefine the operations of "the big 
three" [6] - copy constructor, destructor and the assignment operator. 

The constructor without parameter and the destructor are the same for the two classes Bn_array and 
Bn_tuple. Actually when we work with objects of the class Bn_X, from a mathematical point of view it 
is necessary to point the dimension of the matrix and this dimension does not change. In this aspect using 
a constructor without parameter does not have sense and we are not interested in it. But yet we add such 
a constructor to make our presentation complete and to make, that "the big three" to become "the big 
four "[6]. 

We use the universal integer type int to save the exactness in testing the program, this type can be 
changed to any other whole number type. 

To evaluate the effectiveness and speed of the algorithms, which use objects of the algebra B n (&&, |, !, t(), *) 
it is necessary to evaluate the algorithms, which realize the operations &&, || !, t(), including the operation 
<, comparing two elements, and operation = "assignment". In that sense we describe in details just these 
methods, realizing the above mentioned operations. We suppose that the experienced programmer can easyly 
create the other methods in each of the two classes. 

In the present work we predefine the operator "<", too. Using the same model we can predefine the other 
relational operators: "<=", ">", ">=", "==" and "!=". If the dimensions of the two matrices, we compare, 
are not equal, then the relation "<" is not defined and the result we get is the negative number -1. 

3 Realization of the class Bn_ array. 

When "X"=="array" we propose the following (standard) realization of the examined methods in the class 
Bn_array : 

Bn_array Bn_array : : t () { 
Bn_array temp(n); 
for (int i=0; i<n; i++) 

for (int j=0; j<n; j++) 

*(temp.Matr + i*n+j) = *(Matr + j*n+i); 
return temp ; 

} 
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Bn_array Bn_array : : operator && (Bn_array &B) { 
Bn_array temp(n); 
int n2 = n*n; 
if (B.get_n() != n) 

cout<< "unallowable value of a parameter \n" ; 
else 

for (int p=0; p<n2; p++) 

*(temp.Matr + p) = *(this->Matr + p) && *(B.Matr + p) ; 
return temp ; 

> 

Bn_array Bn_array : : operator I I (Bn_array &B) { 
Bn_array temp(n); 
int n2 = n*n; 
if (B.get_n() != n) 

cout<< "unallowable value of a parameter \n" ; 
else 

for (int p=0; p<n2; p++) 

*(temp.Matr + p) = *(this->Matr + p) II *(B.Matr + p) ; 
return temp ; 

} 

Bn_array Bn_array : : operator ! () { 
int n2 = n*n; 
for (int p=0; p<n2; p++) 

*(this->Matr + p) = *(this->Matr + p) ? : 1; 
return *this; 

> 

Bn_array Bn_array : : operator * (Bn_array &B) { 
Bn_array temp(n); 
int c ; 

if (B.get_n() != n) 

cout<< "unallowable value of a parameter \n" ; 
else 

for (int i=0; i<n; i++) 
for (int j=0; j<n; j++) { 
c=0; 

for (int k=0; k<n; k++) c = c II (* (this->Matr + i*n+k) && *(B.Matr +k*n+j)); 
*(temp.Matr +i*n+j)=c; 

} 

return temp ; 

} 

Bn_array& Bn_array : : operator = (const Bn_array &B) { 
int n2=n*n; 

if (B.get_n() != n) 
cout<< "unallowable value of a parameter \n" ; 
else 

for (int p=0; p<n2; p++) 

*(this->Matr + p) = *(B.Matr + p) ; 
return *this; 
} 

int Bn_array : : operator < (Bn_array &B) { 
int p = ; 
int n2 = n*n; 
if (B.get_n() != n) 

cout<< "unallowable value of a parameter \n" ; 
else 
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while ((*(Matr +p) == *(B.Matr +p)) && (p<n2-l) ) p++; 
if ( *(Matr +p) < *(B.Matr +p) ) return 0; 
else return 1; 

> 

Analogously we can realize the remaining relational operators <=, ==, >, >=, ==, !=. 
It is easy to convince that the following proposition is true: 

Proposition 1 For each whole positive number n, for the computer representation via the class Bn_ array, 
using the C++ programming language, of the algebra $„(&&, ||, !, t(), *), the following propositions are true: 

(i) When we use standard realization (standard predefining) of the operation functions &&, ||, \, <, the 
operation transpose and the assignment operator, each of them does 0{n 2 ) operations of the set Op; 

(ii) When we use standard realization (standard predefining) of the operation function *, this operation 
does 0(n 3 ) operations of the set Op; 

(Hi) For every object of the class Bn_ array are necessary 0(n 2 ) * sizeof (hit) bytes of the operating 
memory; 

(iv) To make initialization of the object of the class Bn_array is done 0{n 2 ) operations of the set Op. 

We can prove the propositions (i) and (ii) as we count the number of the inner cycles and the maximal 
number of the iterations in each of the methods. The propositions (iii) and (iv) are obvious. 

□ 

4 Representing the binary matrices using ordered n-tuples of whole 
nonnegative numbers. 

As it is shown in [5] there is one to one corresponding between the set of all n x n binary matrices and the 
set of all ordered n-tuples of whole numbers, which belong to the closed interval [0, 2™ — 1], based on the 
binary presentation of the whole numbers. This idea takes place in the realization of the class Bn_tuple. 

To create the class Bn_tuple we propose the following methods, which realize the examined operations 
in the algebra $„(&&, ||, !,t(), *). To create these methods we use bitwise operations: bitwise conjunction 
&, bitwise disjunction |, bitwise exclusive "or" A and bitwise negation ~, using these operations we raise the 
effectiveness and make the algorithms work faster. 

Bn_tuple Bn_tuple :: t() { 
Bn_tuple temp(n); 
int k; 

for (int i=0; i<n; i++) 

for (int j=0; j<n; j++) { 

k=get_element (i , j ) ; 

if (k) temp . set_l ( j , i) ; 

else temp . set_0( j , i) ; 

} 

return temp; 
} 

Bn_tuple Bn_tuple : : operator && (Bn_tuple &B) { 
Bn_tuple temp(n); 
if (B.get_n() != n) 

cout<<"unallowable value of a parameter \n" ; 
else 

for (int p=0; p<n; p++) 

*(temp.Matr + p) = *(this->Matr + p) & *(B.Matr + p) ; 
return temp ; 

} 

Bn_tuple Bn_tuple : : operator I I (Bn_tuple &B) ■[ 
Bn_tuple temp(n); 
if (B.get_n() != n) 



5 



cout<<" unallowable value of a parameter \n" ; 
else 

for (int p=0; p<n; p++) 

*(temp.Matr + p) = *(this->Matr + p) I *(B.Matr + p) ; 
return temp; 

} 

Bn_tuple Bn_tuple : : operator ! () { 

Bn_tuple temp(n) ; 

for (int i=0; i<n; i++) { 

for (int j=0; j<n; j++) { 

if ( get_element (i , j ) ) temp . set_0(i , j ) ; 
else temp . set_l (i , j) ; 

} 

} 

return temp; 
} 

Bn_tuple Bn_tuple : : operator * (Bn_tuple &B) { 
Bn_tuple temp(n), TB(n); 
TB = t(B) ; 
int c, r_i, r_j ; 
if (B.get_n() != n) 

cout<< "unallowable value of a parameter \n" ; 

else 

for (int i=0; i<n; i++) 
for (int j=0; j<n; j++) { 
r_i = this->get_row(i) ; 
r_j = TB.get_row(j) ; 
c = r_i & r_j ; 
if (c==0) temp . set_0(i , j ) ; 
else temp . set_l (i , j ) ; 

> 

return temp ; 

} 

Bn_tuple& Bn_tuple : : operator = (const Bn_tuple &B) { 
if (B.get_n() != n) 

c out << "unallowable value\n" ; 
else 

for (int p=0; p<n; p++) 

*(this->Matr + p) = *(B.Matr + p) ; 
return *this; 
} 

int Bn_tuple : : operator < (Bn_tuple &B) { 

int p = ; 

if (B.get_n() ! = n) 

c out << "unallowable value\n" ; 
else 

while ((get_row(p)==B.get_row(p))&&(p<n-l )) p++; 
if (get_row(p) <B . get_row(p) ) return 1; 
else return 0; 

} 

Analogously to proposition [I] we are convinced that the following proposition is true: 

Proposition 2 For each whole positive number n, for the computer representation via the class Bn_ tuple, 
using the C++ programming language, of the algebra £?„(&&, ||, !, t(), *), the following propositions are true: 

(i) When we use the above- described realization of the operation functions && ; ||, < and the predefining 
of the assignment operator, each one of them does 0{n) operations of the set Op; 

(ii) The transpose and negation operation do 0{n 2 ) operations of the set Op. 
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(Hi) When we use the above-described realization of the operation functions *, this operation does 0(n 2 ) 
operations of the set Op; 

(iv) For every object of the class Bn_tuple are necessary 0(n) *sizeof (int) bytes of the operating memory; 

(v) Initialization of the object of the class Bn_tuple can be done using O(n) operations of the set Op. 

We see that to create the class Bn_array is easy and it is not a difficult task even for the beginning pro- 
grammer, when we describe the algorithms we conform with the definitions of the corresponding operations. 
On the other side comparing proposition Q] with proposition [5] we are convinced that following proposition 
is true: 

Theorem 1 The algorithms using objects of the class Bn_tuple work faster than the algorithms using objects 
of the class Bn_ array and they save a lot of operating memory. 
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